1. Set接口概述

一个不包含重复元素的 collection,无序。

哈希表确定元素是否相同

1、 判断的是两个元素的哈希值是否相同。
如果相同,再判断两个对象的内容是否相同。

2、 判断哈希值相同,其实判断的是对象的HashCode方法。判断内容相同,用的是equals方法。

1.1 HashSet类概述

  • 不保证 set 的迭代顺序,特别是它不保证该顺序恒久不变。
  • HashSet如何保证元素唯一性
  • 底层数据结构是哈希表(元素是链表的数组)
  • 哈希表依赖于哈希值存储
  • 添加功能底层依赖两个方法:int hashCode()、boolean equals(Object obj)

HashSet集合之所以能确保不出现重复的元素,是因为它在存入元素时做了很多工作。当调用HashSet集合的add()方法存入元素时,首先调用当前存入对象的hashCode()方法获得对象的哈希值,然后根据对象的哈希值计算出一个存储位置。如果该位置上没有元素,则直接将元素存入,如果该位置上有元素存在,则会调用equals()方法让当前存入的元素依次和该位置上的元素进行比较,如果返回的结果为false就将该元素存入集合,返回的结果为true则说明有重复元素,就将该元素舍弃。整个存储的流程如下图所示。

1500708070711

HashSet存储元素保证唯一性的代码及图解:

Set接口 - 图2

  1. package cn.itcast;
  2. import java.util.HashSet;
  3. class Dog {
  4. private String name;
  5. private int age;
  6. private String color;
  7. private char sex;
  8. public Dog() {
  9. super();
  10. }
  11. public Dog(String name, int age, String color, char sex) {
  12. super();
  13. this.name = name;
  14. this.age = age;
  15. this.color = color;
  16. this.sex = sex;
  17. }
  18. public String getName() {
  19. return name;
  20. }
  21. public void setName(String name) {
  22. this.name = name;
  23. }
  24. public int getAge() {
  25. return age;
  26. }
  27. public void setAge(int age) {
  28. this.age = age;
  29. }
  30. public String getColor() {
  31. return color;
  32. }
  33. public void setColor(String color) {
  34. this.color = color;
  35. }
  36. public char getSex() {
  37. return sex;
  38. }
  39. public void setSex(char sex) {
  40. this.sex = sex;
  41. }
  42. @Override
  43. public int hashCode() {
  44. final int prime = 31;
  45. int result = 1;
  46. result = prime * result + age;
  47. result = prime * result + ((color == null) ? 0 : color.hashCode());
  48. result = prime * result + ((name == null) ? 0 : name.hashCode());
  49. result = prime * result + sex;
  50. return result;
  51. }
  52. @Override
  53. public boolean equals(Object obj) {
  54. if (this == obj)
  55. return true;
  56. if (obj == null)
  57. return false;
  58. if (getClass() != obj.getClass())
  59. return false;
  60. Dog other = (Dog) obj;
  61. if (age != other.age)
  62. return false;
  63. if (color == null) {
  64. if (other.color != null)
  65. return false;
  66. } else if (!color.equals(other.color))
  67. return false;
  68. if (name == null) {
  69. if (other.name != null)
  70. return false;
  71. } else if (!name.equals(other.name))
  72. return false;
  73. if (sex != other.sex)
  74. return false;
  75. return true;
  76. }
  77. }
  78. /*
  79. * HashSet集合存储自定义对象并遍历。如果对象的成员变量值相同即为同一个对象
  80. *
  81. * 注意了: 你使用的是HashSet集合,这个集合的底层是哈希表结构。 而哈希表结构底层依赖:hashCode()和equals()方法。
  82. * 如果你认为对象的成员变量值相同即为同一个对象的话,你就应该重写这两个方法。 如何重写呢?不同担心,自动生成即可。
  83. */
  84. public class DogDemo {
  85. public static void main(String[] args) {
  86. // 创建集合对象
  87. HashSet<Dog> hs = new HashSet<Dog>();
  88. // 创建狗对象
  89. Dog d1 = new Dog("秦桧", 25, "红色", '男');
  90. Dog d2 = new Dog("高俅", 22, "黑色", '女');
  91. Dog d3 = new Dog("秦桧", 25, "红色", '男');
  92. Dog d4 = new Dog("秦桧", 20, "红色", '女');
  93. Dog d5 = new Dog("魏忠贤", 28, "白色", '男');
  94. Dog d6 = new Dog("李莲英", 23, "黄色", '女');
  95. Dog d7 = new Dog("李莲英", 23, "黄色", '女');
  96. Dog d8 = new Dog("李莲英", 23, "黄色", '男');
  97. // 添加元素
  98. hs.add(d1);
  99. hs.add(d2);
  100. hs.add(d3);
  101. hs.add(d4);
  102. hs.add(d5);
  103. hs.add(d6);
  104. hs.add(d7);
  105. hs.add(d8);
  106. // 遍历
  107. for (Dog d : hs) {
  108. System.out.println(d.getName() + "---" + d.getAge() + "---"
  109. + d.getColor() + "---" + d.getSex());
  110. }
  111. }
  112. }

运行结果:

Set接口 - 图3

1.2 LinkedHashSet类概述

元素有序唯一,由链表保证元素有序,由哈希表保证元素唯一。

1.3 TreeSet类概述

使用元素的自然顺序对元素进行排序,或者根据创建 set 时提供的 Comparator 进行排序,具体取决于使用的构造方法。

  • TreeSet是如何保证元素的排序和唯一性
    底层数据结构是红黑树(红黑树是一种自平衡的二叉树)

  • TreeSet判断元素唯一性的方式
    就是根据比较方法的返回结果是否是0,是0,就是相同元素,不存。

  • TreeSet对元素进行排序的方式一
    让元素自身具备比较功能,元素就需要实现Comparable接口,覆盖compareTo方法。
    如果不要按照对象中具备的自然顺序进行排序。如果对象中不具备自然顺序。怎么办?

  • 可以使用TreeSet集合第二种排序方式
    让集合自身具备比较功能,定义一个类实现Comparator接口,覆盖compare方法。将该类对象作为参数传递给TreeSet集合的构造函数。

TreeSet存储元素自然排序和唯一的图解:

Set接口 - 图4

  1. package cn.itcast;
  2. import java.util.Comparator;
  3. import java.util.TreeSet;
  4. class Student {
  5. private String name;
  6. private int age;
  7. public Student() {
  8. super();
  9. }
  10. public Student(String name, int age) {
  11. super();
  12. this.name = name;
  13. this.age = age;
  14. }
  15. public String getName() {
  16. return name;
  17. }
  18. public void setName(String name) {
  19. this.name = name;
  20. }
  21. public int getAge() {
  22. return age;
  23. }
  24. public void setAge(int age) {
  25. this.age = age;
  26. }
  27. }
  28. class MyComparator implements Comparator<Student> {
  29. public int compare(Student s1, Student s2) {
  30. // int num = this.name.length() - s.name.length();
  31. // this -- s1
  32. // s -- s2
  33. // 姓名长度
  34. int num = s1.getName().length() - s2.getName().length();
  35. // 姓名内容
  36. int num2 = num == 0 ? s1.getName().compareTo(s2.getName()) : num;
  37. // 年龄
  38. int num3 = num2 == 0 ? s1.getAge() - s2.getAge() : num2;
  39. return num3;
  40. }
  41. }
  42. /*
  43. * 需求:请按照姓名的长度排序
  44. *
  45. * TreeSet集合保证元素排序和唯一性的原理 唯一性:是根据比较的返回是否是0来决定。 排序: A:自然排序(元素具备比较性)
  46. * 让元素所属的类实现自然排序接口 Comparable B:比较器排序(集合具备比较性) 让集合的构造方法接收一个比较器接口的子类对象 Comparator
  47. */
  48. public class TreeSetDemo {
  49. public static void main(String[] args) {
  50. // 创建集合对象
  51. // TreeSet<Student> ts = new TreeSet<Student>(); //自然排序
  52. // public TreeSet(Comparator comparator) //比较器排序
  53. // TreeSet<Student> ts = new TreeSet<Student>(new MyComparator());
  54. // 如果一个方法的参数是接口,那么真正要的是接口的实现类的对象
  55. // 而匿名内部类就可以实现这个东西
  56. TreeSet<Student> ts = new TreeSet<Student>(new Comparator<Student>() {
  57. public int compare(Student s1, Student s2) {
  58. // 姓名长度
  59. int num = s1.getName().length() - s2.getName().length();
  60. // 姓名内容
  61. int num2 = num == 0 ? s1.getName().compareTo(s2.getName())
  62. : num;
  63. // 年龄
  64. int num3 = num2 == 0 ? s1.getAge() - s2.getAge() : num2;
  65. return num3;
  66. }
  67. });
  68. // 创建元素
  69. Student s1 = new Student("linqingxia", 27);
  70. Student s2 = new Student("zhangguorong", 29);
  71. Student s3 = new Student("wanglihong", 23);
  72. Student s4 = new Student("linqingxia", 27);
  73. Student s5 = new Student("liushishi", 22);
  74. Student s6 = new Student("wuqilong", 40);
  75. Student s7 = new Student("fengqingy", 22);
  76. Student s8 = new Student("linqingxia", 29);
  77. // 添加元素
  78. ts.add(s1);
  79. ts.add(s2);
  80. ts.add(s3);
  81. ts.add(s4);
  82. ts.add(s5);
  83. ts.add(s6);
  84. ts.add(s7);
  85. ts.add(s8);
  86. // 遍历
  87. for (Student s : ts) {
  88. System.out.println(s.getName() + "---" + s.getAge());
  89. }
  90. }
  91. }

运行结果:

Set接口 - 图5